home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 16 / CU Amiga Magazine's Super CD-ROM 16 (1997-10-16)(EMAP Images)(GB)[!][issue 1997-11].iso / CUCD / Graphics / Ghostscript / source / zmisc2.c < prev    next >
C/C++ Source or Header  |  1997-03-08  |  8KB  |  244 lines

  1. /* Copyright (C) 1992, 1995, 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* zmisc2.c */
  20. /* Miscellaneous Level 2 operators */
  21. #include "memory_.h"
  22. #include "string_.h"
  23. #include "ghost.h"
  24. #include "errors.h"
  25. #include "oper.h"
  26. #include "estack.h"
  27. #include "idict.h"
  28. #include "idparam.h"
  29. #include "iparam.h"
  30. #include "dstack.h"
  31. #include "ilevel.h"
  32. #include "iname.h"
  33. #include "iutil2.h"
  34. #include "ivmspace.h"
  35. #include "store.h"
  36.  
  37. /* Forward references */
  38. private int set_language_level(P1(int));
  39.  
  40. /* ------ Language level operators ------ */
  41.  
  42. /* - .languagelevel <1 or 2> */
  43. private int
  44. zlanguagelevel(register os_ptr op)
  45. {    push(1);
  46.     ref_assign(op, &ref_language_level);
  47.     return 0;
  48. }
  49.  
  50. /* <1 or 2> .setlanguagelevel - */
  51. private int
  52. zsetlanguagelevel(register os_ptr op)
  53. {    int code = 0;
  54.     check_type(*op, t_integer);
  55.     if ( op->value.intval < 1 || op->value.intval > 2 )
  56.         return_error(e_rangecheck);
  57.     if ( op->value.intval != ref_language_level.value.intval )
  58.     {    code = set_language_level((int)op->value.intval);
  59.         if ( code < 0 ) return code;
  60.     }
  61.     pop(1);
  62.     ref_assign_old(NULL, &ref_language_level, op, "setlanguagelevel");
  63.     return code;
  64. }
  65.  
  66. /* ------ The 'where' hack ------ */
  67.  
  68. private int
  69. z2where(register os_ptr op)
  70. {    /*
  71.      * Aldus Freehand versions 2.x check for the presence of the
  72.      * setcolor operator, and if it is missing, substitute a procedure.
  73.      * Unfortunately, the procedure takes different parameters from
  74.      * the operator.  As a result, files produced by this application
  75.      * cause an error if the setcolor operator is actually defined
  76.      * and 'bind' is ever used.  Aldus fixed this bug in Freehand 3.0,
  77.      * but there are a lot of files created by the older versions
  78.      * still floating around.  Therefore, at Adobe's suggestion,
  79.      * we implement the following dreadful hack in the 'where' operator:
  80.      *    If the key is /setcolor, and
  81.      *      there is a dictionary named FreeHandDict, and
  82.      *      currentdict is that dictionary,
  83.      *    then "where" consults only that dictionary and not any other
  84.      *      dictionaries on the dictionary stack.
  85.      */
  86.     ref rkns, rfh;
  87.     const ref *pdref = dsp;
  88.     ref *pvalue;
  89.     if ( !r_has_type(op, t_name) ||
  90.          (name_string_ref(op, &rkns), r_size(&rkns)) != 8 ||
  91.          memcmp(rkns.value.bytes, "setcolor", 8) != 0 ||
  92.          name_ref((const byte *)"FreeHandDict", 12, &rfh, -1) < 0 ||
  93.          (pvalue = dict_find_name(&rfh)) == 0 ||
  94.          !obj_eq(pvalue, pdref)
  95.        )
  96.         return zwhere(op);
  97.     check_dict_read(*pdref);
  98.     if ( dict_find(pdref, op, &pvalue) > 0 )
  99.     {    ref_assign(op, pdref);
  100.         push(1);
  101.         make_true(op);
  102.     }
  103.     else
  104.         make_false(op);
  105.     return 0;
  106. }
  107.  
  108. /* ------ Initialization procedure ------ */
  109.  
  110. /* The level setting ops are recognized even in Level 1 mode. */
  111. BEGIN_OP_DEFS(zmisc2_op_defs) {
  112.     {"0.languagelevel", zlanguagelevel},
  113.     {"1.setlanguagelevel", zsetlanguagelevel},
  114.         /* The rest of the operators are defined only in Level 2. */
  115.         op_def_begin_level2(),
  116. /* Note that this overrides the definition in zdict.c. */
  117.     {"1where", z2where},
  118. END_OP_DEFS(0) }
  119.  
  120. /* ------ Internal procedures ------ */
  121.  
  122. /* Adjust the interpreter for a change in language level. */
  123. /* This is used for the .setlanguagelevel operator, */
  124. /* and after a restore. */
  125. private int swap_entry(P3(ref elt[2], ref *pdict, ref *pdict2));
  126. private int
  127. set_language_level(int level)
  128. {    ref *pgdict =        /* globaldict, if present */
  129.       ref_stack_index(&d_stack, ref_stack_count(&d_stack) - 2);
  130.     ref *level2dict;
  131.     if (dict_find_string(systemdict, "level2dict", &level2dict) <= 0)
  132.           return_error(e_undefined);          
  133.     /* As noted in dstack.h, we allocate the extra d-stack entry */
  134.     /* for globaldict even in Level 1 mode; in Level 1 mode, */
  135.     /* this entry holds an extra copy of systemdict, and */
  136.     /* [count]dictstack omit the very bottommost entry. */
  137.     if ( level == 2 )        /* from Level 1 to Level 2 */
  138.     {    /* Put globaldict in the dictionary stack. */
  139.         ref *pdict;
  140.         int code = dict_find_string(level2dict, "globaldict", &pdict);
  141.  
  142.         /* This might be called so early in initialization that */
  143.         /* globaldict hasn't been defined yet.  If so, just skip */
  144.         /* this step. */
  145.         if ( code > 0 )
  146.           { if ( !r_has_type(pdict, t_dictionary) )
  147.               return_error(e_typecheck);
  148.             *pgdict = *pdict;
  149.           }
  150.         /* Set other flags for Level 2 operation. */
  151.         dict_auto_expand = true;
  152.     }
  153.     else                /* from Level 2 to Level 1 */
  154.     {    /* Clear the cached definition pointers of all names */
  155.         /* defined in globaldict.  This will slow down */
  156.         /* future lookups, but we don't care. */
  157.         int index = dict_first(pgdict);
  158.         ref elt[2];
  159.         while ( (index = dict_next(pgdict, index, &elt[0])) >= 0 )
  160.           if ( r_has_type(&elt[0], t_name) )
  161.             name_invalidate_value_cache(&elt[0]);
  162.         /* Overwrite globaldict in the dictionary stack. */
  163.         *pgdict = *systemdict;
  164.         /* Set other flags for Level 1 operation. */
  165.         dict_auto_expand = false;
  166.     }
  167.     /* Swap the contents of level2dict and systemdict. */
  168.     /* If a value in level2dict is a dictionary, and it contains */
  169.     /* a key/value pair referring to itself, swap its contents */
  170.     /* with the contents of the same dictionary in systemdict. */
  171.     /* (This is a hack to swap the contents of statusdict.) */
  172.     {    int index = dict_first(level2dict);
  173.         ref elt[2];        /* key, value */
  174.         ref *subdict;
  175.         while ( (index = dict_next(level2dict, index, &elt[0])) >= 0 )
  176.           if ( r_has_type(&elt[1], t_dictionary) &&
  177.                dict_find(&elt[1], &elt[0], &subdict) > 0 &&
  178.                obj_eq(&elt[1], subdict)
  179.              )
  180.         {
  181. #define sub2dict &elt[1]
  182.             int isub = dict_first(sub2dict);
  183.             ref subelt[2];
  184.             int found = dict_find(systemdict, &elt[0], &subdict);
  185.             if ( found <= 0 )
  186.               continue;
  187.             while ( (isub = dict_next(sub2dict, isub, &subelt[0])) >= 0 )
  188.               if ( !obj_eq(&subelt[0], &elt[0]) )    /* don't swap dict itself */
  189.             {    int code = swap_entry(subelt, subdict, sub2dict);
  190.                 if ( code < 0 )
  191.                   return code;
  192.             }
  193. #undef sub2dict
  194.         }
  195.           else
  196.         {    int code = swap_entry(elt, systemdict, level2dict);
  197.             if ( code < 0 )
  198.               return code;
  199.         }
  200.     }
  201.     dict_set_top();        /* reload dict stack cache */
  202.     return 0;
  203. }
  204.  
  205. /* Swap an entry from a Level 2 dictionary into a base dictionary. */
  206. /* elt[0] is the key, elt[1] is the value in the Level 2 dictionary. */
  207. private int
  208. swap_entry(ref elt[2], ref *pdict, ref *pdict2)
  209. {    ref *pvalue;
  210.     ref old_value;
  211.     int found = dict_find(pdict, &elt[0], &pvalue);
  212.     switch ( found )
  213.     {
  214.     default:        /* <0, error */
  215.         return found;
  216.     case 0:            /* missing */
  217.         make_null(&old_value);
  218.         break;
  219.     case 1:            /* present */
  220.         old_value = *pvalue;
  221.     }
  222.     /*
  223.      * Temporarily flag the dictionaries as local, so that we don't
  224.      * get invalidaccess errors.  (We know that they are both
  225.      * referenced from systemdict, so they are allowed to reference
  226.      * local objects even if they are global.)
  227.      */
  228.     {    uint space2 = r_space(pdict2);
  229.         int code;
  230.         r_set_space(pdict2, avm_local);
  231.         dict_put(pdict2, &elt[0], &old_value);
  232.         if ( r_has_type(&elt[1], t_null) )
  233.             code = dict_undef(pdict, &elt[0]);
  234.         else
  235.         {    uint space = r_space(pdict);
  236.             r_set_space(pdict, avm_local);
  237.             code = dict_put(pdict, &elt[0], &elt[1]);
  238.             r_set_space(pdict, space);
  239.         }
  240.         r_set_space(pdict2, space2);
  241.         return code;
  242.     }
  243. }
  244.